# **CPU** Project

Group Members: Kaleb Badgett, Seth McDaniel, Ivan Ortiz, and Joel Zell

We created a 4 bit cpu with the objective of computing multiplication between two numbers (0-10). It had to be able to complete four different functions: Addition, Subtraction, Exclusive OR, and Move. This is the documentation of this project.

Arithmetic Logic Unit Immediate 4 Operand 1 SRC -Result Operand 2 Opcode DST Opcode **Register File Register** A AND **Register B** AND Clock -

Below is the schematic:

\*Note there should be an inverter before one of the and gates (between DST and the AND gate)

Two Muxes, 2 PLDs, 2 octal flip flop chips, 3 hex inverters, and one AND chip were used in the physical implementation of the project.

In this project, we created machine code using operation (op) code, the source bit, the destination bit, and the immediate value, in that order. This 4-bit processor took an 8-bit input and used that to create the desired outputs. The binary was then converted to hexadecimal to be used as machine code. To imitate a ROM to use our machine code, we used an Arduino.

The op code is as follows: 00 ADD 01 SUB 10 XOR 11 MOV

For example, 1100 1101 became 0xCD. This meant that the value D (12) was to be stored in register A.

Below, you will find a photo of our final design.



#### Muxes:

| Ā/B   | 1 | U 16 | Vcc  |
|-------|---|------|------|
| 1A [  | 2 | 15   | ĪG   |
| 1B [  | 3 | 14   | ] 4A |
| 1Y [  | 4 | 13   | ] 4B |
| 2A [  | 5 | 12   | ] 4Y |
| 2B [  | 6 | 11   | ] 3A |
| 2Y [  | 7 | 10   | ] 3B |
| GND [ | 8 | 9    | ] 3Y |
|       | _ |      |      |

What is high and low?

On the first mux that used selector SRC, 0 selected Immediate, and 1 selected register B. On the second mux that used DST, 0 selected register A, and 1 selected register B.

What pins are what bit?

Both Muxes:

Pin 1 (!A/B) was the selector on both muxes.

Pins 4, 7, 9, and 12 (1Y, 2Y, 3Y, and 4Y respectively) were the outputs of the muxes that led to the hex inverters.

Pin 8 (GND) was the ground pin for both muxes.

Pin 16 (VCC) was the power pin for both muxes.

Pin 15 (!G) was the enable signal that was connected to ground.

Mux 1:

Pins 2, 5, 11, and 14 (1A, 2A, 3A, and 4A respectively) were all for the immediate value. Pins 3, 6, 10, and 13 (1B, 2B, 3B, and 4B respectively) were all for the values coming from register B.

Mux 2:

Pins 2, 5, 11, and 14 (1A, 2A, 3A, and 4A respectively) were all for the values coming from register A.

Pins 3, 6, 10, and 13 (1B, 2B, 3B, and 4B respectively) were all for the values coming from register B.

#### Hex Inverters:

| 1A [<br>1Y [<br>2A [<br>3A [<br>3Y [ | 1<br>2<br>3<br>4<br>5<br>6<br>7 | 14<br>13<br>12<br>11<br>10<br>9 | Vcc<br>6A<br>6Y<br>5A<br>5Y<br>4A |
|--------------------------------------|---------------------------------|---------------------------------|-----------------------------------|
| GND [                                | 7                               | 8                               | 1 4Y                              |

Both Inverters: Pin 7 (GND) was the ground pin. Pin 14 (VCC) was the power pin.

Inverter 1:

Pins 1, 3, 5, and 13 (1A, 2A, 3A, and 6A respectively) were the input pins that took the 4-bit values from mux 1 and inverted them in order to make those values usable later on. Pins 2, 4, 6, and 12 (1Y, 2Y, 3Y, and 6Y respectively) were the outputs of the inverter that led to the ALU (PLDs).

#### Inverter 2:

Pins 1, 3, 5, and 13 (1A, 2A, 3A, and 6A respectively) were the input pins that took the 4-bit values from mux 2 and inverted them in order to make those values usable later on. Pins 2, 4, 6, and 12 (1Y, 2Y, 3Y, and 6Y respectively) were the outputs of the inverter that led to the ALU (PLDs).

Inverter 3 (AND gate): Pin 1 (1A) was the input from DST. Pin 2 (1Y) was the output that led to the AND gate for later use.

## PLDs (ALU):

|           | 24 🗆 VO  |
|-----------|----------|
| IN 🖂 2    | 23 🗔 1/0 |
| IN 🗔 3    | 22 1/0   |
| IN/PD C 4 | 21 1/0   |
| IN 🗔 5    | 20 🖂 1/0 |
| IN 🖂 6    | 19 🗔 I/O |
| IN 🗔 7    | 18 🗔 1/0 |
| IN 🗔 8    | 17 🗔 I/O |
| IN 🗔 9    | 16 🗔 I/O |
| IN 🗔 10   | 15 🗔 I/O |
| IN 🗔 11   | 14 🖂 1/0 |
| GND - 12  | 13 🗔 IN  |

Adder/Subtractor PLD:

Pin 12 (GND) is the ground pin.

Pin 24 (VCC) is the power pin.

Pins 1-10 were the input pins for the opcode (2 bits, pins 1 and 2), the mux 1 values (4 bits, pins 3-6), and the mux 2 values (4 bits, pins 7-10).

Pins 14-17 were the output pins (4 bits) that led to the second PLD.

Pins 18-21 were our pin nodes that acted as our carryout's for the addition.

| ണ്ട്ര | NIW/: | CUPL  | WINC  | UPL\ADDEF | SUBALUPLD.PLD     |            | - |    | 23   |
|-------|-------|-------|-------|-----------|-------------------|------------|---|----|------|
| Name  |       | CPU   | DESIG | IN ;      |                   |            |   |    |      |
| Part  | No    | 00;   |       |           |                   |            |   |    | 1    |
| Date  |       | 4/22  | /2024 | 1 7       |                   |            |   |    | - 11 |
| Revi  | sion  | 01 ;  |       |           |                   |            |   |    | - 11 |
| Desi  | gner  | Engi  | neer  | ;         |                   |            |   |    | - 11 |
| Comp  | any   | Texa  | s Tec | :h ;      |                   |            |   |    | - 11 |
| Asse  | mbly  | None  | ;     |           |                   |            |   |    | - 11 |
| Loca  | tion  | ;     |       |           |                   |            |   |    | - 11 |
| Devi  | ce    | g22v  | 10 ;  |           |                   |            |   |    | - 1  |
| /* *  | ****  | ***** | ***** | INPUT P   | INS ************* | ********/  |   |    |      |
| PIN   | 1     | =     | opt   | )         | ; /*              |            |   | *  | 1    |
| PIN   | 2     | =     | opl   |           | ; /*              |            |   | *  | 1    |
| PIN   | 3     | =     | aO    |           | ; /*              |            |   | */ |      |
| PIN   | 4     | =     | al    |           | ; /*              |            |   | */ |      |
| PIN   | 5     | =     | a2    |           | ; /*              |            |   | */ |      |
| PIN   | e     | =     | a3    |           | ; /*              |            |   | */ |      |
| PIN   | 7     | =     | bO    |           | ; 1*              |            |   | */ |      |
| PIN   | 8     | =     | bl    |           | ; /*              |            |   | *1 |      |
| PIN   | 9     | =     | b2    |           | : /*              |            |   | *1 |      |
| PIN   | 10    | =     | b3    |           | ; /*              |            |   | *  | 1    |
| 1 * * | ****  | ****  | ***** | OUTPUT    | PINS ************ | ********/  |   |    |      |
| PIN   | 14    | =     | fO    | )         | ; /*              |            |   | */ | 1    |
| PIN   | 15    | =     | fl    |           | ; /*              |            |   | *  | 1    |
| PIN   | 16    | =     | £2    | 1         | ; /*              |            |   |    | 1    |
| PIN   | 17    | =     | f3    | 3         | ; /*              |            |   | *  | 1    |
| 1 * * | ****  | ****  | ****  | PINNODE   | S *********       | *********/ |   |    |      |
| PINN  | IODE  | 18    | =     | c0        | ; /*              |            |   | *  | 1    |
| PINN  | IODE  | 19    | =     | cl        | ; /*              |            |   |    |      |
| PINN  | IODE  | 20    | =     | c2        | ; /*              |            |   |    |      |
| PINN  | ODE   | 21    | =     | c3        | ; /*              |            |   |    |      |

```
/* Logic */
c0 = op0;
c1 = (a0 & (b0 $ op0)) $ (a0 & c0) $ ((b0 $ op0) & c0);
c2 = (a1 & (b1 $ op0)) $ (a1 & c1) $ ((b1 $ op0) & c1);
c3 = (a2 & (b2 $ op0)) $ (a2 & c2) $ ((b2 $ op0) & c2);
f0 = ((!op0 & !op1) & (a0 $ b0 $ c0)) $ ((op0 & !op1) & (a0 $ (b0 $ op0) $
c0));
f1 = ((!op0 & !op1) & (a1 $ b1 $ c1)) $ ((op0 & !op1) & (a1 $ (b1 $ op0) $
c1));
f2 = ((!op0 & !op1) & (a2 $ b2 $ c2)) $ ((op0 & !op1) & (a1 $ (b1 $ op0) $
c2);
f3 = ((!op0 & !op1) & (a3 $ b3 $ c3)) $ ((op0 & !op1) & (a3 $ (b3 $ op0) $
c3));
```

Waterfall PLD (XOR and MOV):

Pin 12 (GND) is the ground pin.

Pin 24 (VCC) is the power pin.

Pins 1-15 were the input pins for the opcode (2-bits, pins 1 and 2), the mux 1 values (4-bits, pins 3-6), the mux 2 values (4-bits, pins 7-10), and the Adder/Subtractor PLD values (4-bits, pins 11-15).

Pins 16-19 were the output pins (4-bits) that led to the registers.

```
C:\WINCUPL\WINCUPL\WATERFALL.PLD
                                                    -
       WATERFALL ;
Name
      00 ;
PartNo
Date
      4/24/2024 ;
Revision 01 ;
Designer Engineer ;
Company Texas Tech ;
Assembly None ;
Location ;
Device g22v10 ;
PIN 1
                             ; /*
                                                          *
      =
         0go
PIN 2
                             7 /*
      =
          opl
                                                          *
PIN 3
                            : 1*
                                                         */
      =
          a0
PIN 4
      =
          al
                            ; /*
                                                         */
PIN 5
      =
          a2
                            ; 1*
                                                         */
PIN 6
      =
          a3
                            ; /*
                                                         */
PIN 7
      =
          bO
                            : 1*
                                                         *1
PIN 8
      =
         bl
                            1 /*
                                                         */
                            ; /*
PIN 9
      =
          b2
                                                         *1
PIN 10 =
          b3
                            ; /*
                                                         *1
      =
PIN
   11
          addsub0
                                ; /*
                                ; /*
PIN
   13 =
          addsubl
                                ; /*
PIN 14 =
          addsub2
                                ; /*
PIN 15 =
         addsub3
PIN 16 = f0
                            ; 10
                                                         *1
                            ; /*
PIN 17 =
                                                         */
         f1
PIN 18 =
                            : 14
                                                         */
         f2
                                                         w.f
PIN 19 = f3
                            ; /*
```

#### AND Gate:

| 1A 🖂 | 10 | 14 |      |
|------|----|----|------|
| 1B 🗖 | 2  | 13 | 1 4B |
| 1Y 🗖 | 3  | 12 | 4A   |
| 2A 💶 | 4  | 11 | 1 4Y |
| 2B 💶 | 5  | 10 |      |
| 2Y 🖵 | 6  | 9  |      |
| GND  | 7  | 8  | 1 3Y |
|      |    |    |      |

Pin 7 (GND) is the ground pin.

Pin 14 (VCC) is the power pin.

Pins 1, 2, 4, and 5 (1A, 1B, 2A, and 2B respectively) were the inputs that took DST, !DST, and CLK.

Pins 3 and 6 (1Y, and 2Y respectively) were the output pins that led to the clock signals of the registers.

### D Flip Flops:

| OE [ | 1                                        | 20 | Vcc |
|------|------------------------------------------|----|-----|
| 1D   | 2                                        | 19 | 10  |
| 2D [ | 3                                        | 18 | 2Q  |
| 3D   | 4                                        | 17 | 3Q  |
| 4D [ | 5                                        | 16 | 4Q  |
| 5D [ | 6                                        | 15 | 5Q  |
| 6D   | 7                                        | 14 | 6Q  |
| 7D [ | 8                                        | 13 | 7Q  |
| 8D   | 9                                        | 12 | 8Q  |
| GND  | 10                                       | 11 | CLK |
|      | C. C |    |     |

Register A:

Pin 20 (VCC) was the power pin.

Pin 10 (GND) was the ground pin.

Pin 11 (CLK) was the clock signal.

Pin 1 (!OE) is an enable signal that was grounded.

Pins 2, 3, 4, and 5 (1D, 2D, 3D, and 4D respectively) were the inputs from the ALU that were to be stored as the register A values.

Pins 19, 18, 17, and 16 (1Q, 2Q, 3Q, and 4Q respectively) were the output pins that went to the green LEDs and back to the second mux.

Register B:
Pin 20 (VCC) was the power pin.
Pin 10 (GND) was the ground pin.
Pin 11 (CLK) was the clock signal.
Pin 1 (!OE) was an enable signal that was grounded.
Pins 2, 3, 4, and 5 (1D, 2D, 3D, and 4D respectively) were the inputs from the ALU that were to be stored as the register B values.
Pins 19, 18, 17, and 16 (1Q, 2Q, 3Q, and 4Q respectively) were the output pins that went to the

## Wire Coloring:

Color organization varied by section due to lack of wires of required length.

## Arduino (ROM) Code:

red LEDs and back to both muxes.

Below is the code we used for a subtraction computation.

//My pin setup
int pins[]={6,7,8,9,10,11,12,13};// I/O pins
int clock = 5;
//9,7,8,6 --> immediate
//13 -> OP1
//12 -> OP2
//11 -> SRC
//10 -> DST
void setup(){
pinMode(clock,OUTPUT);
for(int i=0; i<8; i++)
{
 pinMode(pins[i], OUTPUT);
}</pre>

```
void loop() {
```

int program\_counter = 0;

byte rom[]= {

0xC4,

0xD5,

0x43,

0x54,

## **};**

for (int i=0; i<4;i++)

# {

```
for(int j=0; j<8; j++)</pre>
```

# {

digitalWrite(pins[j], bitRead(rom[program\_counter],j));

## }

```
program_counter++;
```

delay(1000);

```
digitalWrite(clock,1);
```

delay(1000);

```
digitalWrite(clock,0);
```

delay(1000);

## }

```
delay(60000000);
```

Below is the code we used in our multiplication algorithm. We multiplied 5x9.

| //My pin setup                               |  |
|----------------------------------------------|--|
| int pins[]={6,7,8,9,10,11,12,13};// I/O pins |  |
| int clock = 5;                               |  |
| //9,7,8,6> immediate                         |  |
| //13 - > OP1                                 |  |
| //12 - > OP2                                 |  |
| //11 - > SRC                                 |  |
| //10 - > DST                                 |  |
| void setup(){                                |  |
| pinMode(clock,OUTPUT);                       |  |
| for(int i=0; i<8; i++)                       |  |
| {                                            |  |
| pinMode(pins[i], OUTPUT);                    |  |
| }                                            |  |
| }                                            |  |
|                                              |  |
| void loop() {                                |  |
| int program_counter = 0;                     |  |
| byte rom[]= {                                |  |
| 0xC0,                                        |  |
| 0xD0,                                        |  |
| 0x15,                                        |  |
|                                              |  |

```
0x00,
 0x10,
 0x00,
 0x18,
 0x00,
 0xE0,//remove this and reverse wires****this moves B to A
 0xD0,
 I*remove this and reverse wires****this moves the value of A "0" to the register,
 this value is stored in our heads because we can not switch the two in one clock cycle,
 the outputs from reg b to output a and reg a to output b(current output has a going to "a"
 and reg b going to "b")*/
 0x10,
 0x12,
};
for (int i=0; i<12;i++)
{
  for(int j=0; j<8; j++)
  {
   digitalWrite(pins[j], bitRead(rom[program_counter],j));
  }
  program_counter++;
  delay(1000);
  digitalWrite(clock,1);
  delay(1000);
```

digitalWrite(clock,0);

delay(1000);

}

delay(60000000);

#### **Overall:**

What skills were used?

We had to use technical skills, active listening, data analysis, problem solving, computer skills, teamwork, programming, communication, and organizational skills throughout the duration of this project.

What struggles did we run into?

We ran into several issues while completing this project.

1. Incorrect wiring.

Upon attempting the first few trials using our CPU, three of our functions would not work due to the enable signal on our muxes not being grounded. We had to search the breadboard for issues and fix this.

2. Breadboard malfunction.

We worked for a number of days on this project, but kept running into the same problem. Our results were not correct. We tried to troubleshoot many possible issues, but after several days, Seth McDaniel found the issue. One of the rails on one of the breadboards was out and would not transfer any sort of signal. Once Seth found this problem, he fixed it and we were able to resume work the following day.

3. PLD program error.

After fixing the error with the breadboard, some of our functions still were not working properly. We found an error in the wincupl code used on one of the PLDs and fixed it.

Ensuring machine code was correct.
 One error we had was within the machine code we wrote for the arduino. After further investigation, we were able to fix this problem.

Did we have a group leader?

We did not have a designated group leader.

Did we need a group leader?

While having a designated group leader may have helped when it came to project management, we were able to complete all project requirements by ensuring each of us aided wherever and whenever needed.

What did we learn from the project?

Through this project, we learned about the inner workings of a CPU, a ROM, and an ALU. As we faced different issues we learned how to overcome those problems, which then gave us a deeper understanding of the parts we were working with.

What did we learn from working as a group?

While working as a group, we learned how to divide responsibilities to ensure we met the schedule and even completed the project a couple days early. We also learned how to overcome issues as a group rather than individually. This was done by ensuring multiple people were working on the project at once which, for the most part, allowed us to find issues quicker than we may have otherwise.